





# Computer Organization & Design 实验与课程设计

实验九

# 多周期IP核集成CPU

--建立多周期CPU调试、测试和应用环境

施青松

Asso. Prof. Shi Qingsong
College of Computer Science and Technology, Zhejiang University
zjsqs@zju.edu.cn

## **Course Outline**



# 实验目的与实验环境

实验任务

实验原理

实验操作与实现

淅沙人学系统结构与系统软件实验室

# 实验目的



- 1. 深入理解CPU结构
- 3. 学习CPU性能优化:多周期
- 3. 建立多周期CPU测试应用环境
- 4. IP核深入应用

# 实验环境



#### □实验设备

- 1. 计算机(Intel Core i5以上,4GB内存以上)系统
- 2. Spartan-3 Starter Kit Board/Sword开发板
- 3. Xilinx ISE14.4及以上开发工具

#### □材料

无

# **Course Outline**



实验目的与实验环境

实验任务

实验原理

实验操作与实现

洲沙太学系统结构与系统软件实验室

# 实验任务



- 1. 搭建多周期CPU测试应用环境
  - ■用结构描述重建Exp03顶层模块
    - □用多周期CPU核替换单周期
- 2. 用数据通路和控制器核集成替换CPU核
  - □除CPU外复用Exp03的部件模块

## **Course Outline**



实验目的与实验环境

实验任务

实验原理

实验操作与实现

浙江大学系统结构与系统软件实验室

### **CPU** organization



#### **□** Digital circuit

General circuits that controls logical event with logical gates Hardware



#### **□** Computer organization

Special circuits that processes logical action with instructions
 -Software



# **Computer Organization**



#### **□** Decomposability of computer systems



# 多周期处理器测试框架或SOC





浙江大学 计算机学院 系统结构与系统软件实验室

ZheJiang Univers

# 系统分解为九个子模块



- □用此9个模块,用结构描述建立SOC测试构架
- □集成实现多周期处理器SOC

U1: **MCPU** 

U3: **RAM** 

总线(含外设3~4) ■ U4:

7段显示接口 ■ U5:

外设1-7段显示设备 ■ U6:

外设2-GPIO接口及LED ■ U7:

辅助模块一,通用分频模块 U8:

U9: 辅助模块二,机械去抖模块

通用计数器 U10:

-Muliti-CPU

-RAM\_B

-MIO\_BUS

-Multi\_8CH.

-SSeg\_Dev

-SPI

-clk\_div

-SAnti\_jitter

-Counter\_x

□ 以上除U1外与单周期共享

浙江大学 计算机学院 系统结构与系统软件实验室

OExp09-IP2MCPU xc7k160t-2Lffg676

🖃 🗗 🥌 Top\_OExp09\_IP2MCPU (Top\_OExp09\_IP2MCPU.sch

U61 - Seg7 Dev (Seg7 Dev.ngc)

V U61 - Seg7 Dev (Seg7 Dev IO.v)

U5 - Multi\_8CH32 (Multi\_8CH32.ngc)

V U5 - Multi 8CH32 (Multi 8CH32 IO.v)

V U8 - clk div (clk div.v)

U7 - SPIO (SPIO.ngc)

V U7 - SPIO (SPIO IO.v)

U6 - SSeg7 Dev (SSeg7 Dev.ngc)

V U6 - SSeg7 Dev (SSeg7 Dev IO.v)

U9 - SAnti jitter (SAnti jitter.ngc)

V U9 - SAnti\_jitter (SAnti\_jitter\_IO.v)

M4 - SEnter\_2\_32 (SEnter\_2\_32.ngc)

M4 - SEnter 2 32 (SEnter 2 32 IO.v)

U71 - PIO (PIO.ngc)

V U71 - PIO (PIO IO.v)

U3 - RAM B (RAM B.xco)

U4 - MIO BUS (MIO BUS.ngc)

V U4 - MIO BUS (MIO BUS IO.v)

U10 - Counter\_x (Counter\_x.ngc)

U10 - Counter x (Counter 3 IO.v)

⊕ U1 - Multi\_CPU (Multi\_CPU.sch)

Org-Sword.ucf

# U1-多周期CPU模块: Multi-CPU



- □ MIPS 构架
  - RISC体系结构
  - ⊙ 三种指令类型
- □SOC测试模块的处理核心
  - ■由数据通路和控制器二个核组成
    - □本实验直接用2个IP核集成
    - □本实验也可先用CPU IP核
      - 调试通过后再用2个IP集成
  - 本实验可用IP Core- U1
    - ■核调用模块Multi\_CPU.ngc
    - □核接口信号模块(空文档): Multi\_CPU.v
    - □核模块符号文档: Multi\_SCPU.sym



# CPU核接口空模块-Multi\_CPU.v



```
Muliti_CPU (input wire clk,
module
                                               注意与单周期区别
                     input wire reset,
                     input wire MIO_ready, // be used: =1
                     output wire[31:0]PC_out,//Test
                     output[31:0] inst_out, //TEST
                     output wire mem_w,  //存储器读写控制
                     output wire[31:0]Addr_out,//数据空间访问地址
                     output wire[31:0]Data_out//数据输出总线
                     input wire [31:0]Data_in, //数据输入总线
                     output wire CPU_MIO, // Be used
                     input wire INT //中断
                     output[4:0]state
                                      //Test
                     );
```

endmodule

# CPU部件之一-数据通路: M\_datapath



#### MUX选择更多输入以兼容扩展

# □ 本实验用IP 软核- M\_datapath

- 核调用模块M\_atapath.ngc
- 核接口信号模块: M\_datapath.v
- 核模块符号文档: M\_datapath.sym

#### □重要信号

- Inst: 指令寄存器输出
- PC\_Current: 当前PC(PC+4)
- M\_addr: 存储器地址
- Branch(教材中的beq):
  - $\square = 1$ : beq
  - $\square = 0$ : bne
- PCWriteCond: Branch指令





# 数据通路空模块- M\_datapath.v



| module    | M_datapath(input clk,                             |                 | clk           | zero             |
|-----------|---------------------------------------------------|-----------------|---------------|------------------|
|           | input reset,                                      |                 |               |                  |
|           | <b>input</b> MIO_ready,                           | //=1            | reset         |                  |
|           | input IorD,                                       |                 | MIO_ready     | U1_2             |
|           | input IRWrite,                                    | / マエ (カ 大山 4 八) | le-D          |                  |
|           | input[1:0] RegDst,                                | //预留到2位         | lorD          | overflow ———     |
|           | <pre>input RegWrite, input[1:0]MemtoReg,</pre>    | //预留到2位         | IRWrite N./I  | datapath         |
|           | input ALUSrcA,                                    | //J/A 2/2 E     | RegWrite      | _datapath        |
|           | <pre>input[1:0]ALUSrcB, input[1:0]PCSource,</pre> | //4选1控制         | ALUSrcA       | PC_Current(31:0) |
|           | input PCWrite,                                    |                 | PCWrite       |                  |
|           | <pre>input PCWriteCond, input Branch,</pre>       |                 | PCWriteCond   |                  |
|           | input[2:0]ALU_operation,                          |                 | Branch        | Inst(31:0)       |
|           | output[31:0]PC_Current,                           |                 | RegDst(1:0)   |                  |
|           | <pre>input[31:0]data2CPU, output[31:0]Inst,</pre> |                 | MemtoReg(1:0  | )                |
|           | output[31:0]data_out,                             |                 | ALUSrcB(1:0)  | data_out(31:0)   |
|           | <pre>output[31:0]M_addr, output zero,</pre>       |                 | PCSource(1:0) | )                |
|           | output overflow                                   |                 | ALU_operation | (2:0)            |
| endmodule | );<br>e                                           |                 | data2CPU(31:0 | 0) M_addr(31:0)  |

浙沙大学 计算机学院 系统结构与系统软件实验室

# CPU部件之二-控制器: ctrl



#### □ 本实验用IP 软核- ctrl

- 核调用模块ctrl.ngc
- 核接口信号模块(空文档): ctrl.v
- 核模块符号文档: ctrl.sym

#### □重要信号

- MIO\_ready: 外设就绪
  - □=0 CPU等待
  - □=1 CPU正常运行
  - □本实验恒等于1
- Inst\_in: 指令输入,来自IR输出
- State\_out: 状态编码,用于测试





# 控制器接口文档-ctrl.v



```
module
           ctrl(input clk,
                                                                      clk
                                                                                    MemRead
                input reset.
                                                                                    MemWrite
                input [31:0] Inst_in,
                                                                            U11
                input zero,
                                                                                    CPU MIO
                input overflow,
                input MIO_ready,
                                                                                         IorD
                                                                      reset
                output reg MemRead,
                                                                                      IRWrite
                output reg MemWrite,
                output reg[2:0]ALU_operation,
                                                                                     RegWrite
                output [4:0]state out,
                                                                                     ALUSTCA
                                                                      zero
                output reg CPU MIO,
                                                                                      PCWrite
                                                                           ctrl
                output reg IorD,
                                                                                  PCWriteCond
                output reg IRWrite,
                output reg [1:0]RegDst,
                                                                      overflow
                                                                                       Branch
                output reg RegWrite,
                                                                                   RegDst(1:0)
                output reg [1:0]MemtoReg,
                output reg ALUSrcA,
                                                                                 MemtoReg(1:0)
                output reg [1:0]ALUSrcB,
                output reg [1:0]PCSource,
                                                                      MIO ready
                                                                                  ALUSrcB(1:0)
                output reg PCWrite,
                                                                                 PCSource(1:0)
                output reg PCWriteCond,
                output reg Branch
                                                                              ALU_operation(2:0)
                                                                      Inst_in(31:0)
                                                                                  state_out(4:0)
endmodule
```

浙江大学

计算机学院 系统结构与系统软件实验室

# U3-指令代码存储模块: RAM\_B



### $\square$ RAM B

用Distributed Memory Generator没有clk信号 请编辑删除clka引脚。SP3平台用不用

- 将Lab3的ROM和RAM合并
  - □ 数据代码存储共享
- FPGA内部存储器
  - Block Memory Generator 或Distributed Memory Generator
- 容量与Lab3的RAM\_B相同
  - $\square$  1024  $\times$  32bit
- 核模块符号文档: RAM\_B.sym
  - □ 自动生成符号不规则,需要修整

# □本实验需要重新生成IP固核

- RAM初始化文档: mem.coe
  - □ 代码与数据合并在一个存储器中
- 核调用模块RAM B.xco





□生成后自动调用关联,不需要空文档 大学 计算机学院 系统结构与系统软件实验室

# RAM\_B调用方式:与Lab3相同



#### □ ROM调用接口信号

#### 红色与单周期不同均通过MIO\_BUS

- □ 图形输入调用
  - RAM\_B.sym
- □固核调用不需要空模块文档

# U3-存储器初始化数据文档: mem.coe 代码与数据共存



#### memory\_initialization\_radix=16;

#### memory\_initialization\_vector=

· 代码区: 地址从00000000开始

```
f000000, 000002AB, 8000000, 0000003F, 00000001, FFFF0000, 0000FFFF, 80000000, 00000000, 11111111, 22222222, 33333333, 444444444, 555555555, 666666666, 77777777, 88888888, 99999999, aaaaaaaa, bbbbbbbb, ccccccc, dddddddd, eeeeeeee, ffffffff, 557EF7E0, D7BDFBD9, D7DBFDB9, DFCFFCFB, DFCFBFFF, F7F3DFFF, FFFFDF3D, FFFF9DB9, FFFFBCFB, DFCFFCFB, DFCFBFFF, D7DB9FFF, D7DBFDB9, D7BDFBD9, FFFF07E0, 007E0FFF, 03bdf020, 03def820, 08002300;
```

数据区: 地址起始需要约定



# U4-总线接口模块: MIO\_BUS



#### **■ MIO\_BUS**

- CPU与外部数据交换接口模块
- ■本课程实验将数据交换电路合并成一个模块
  - □非常简单,但非标准,扩展不方便
  - □后继课程采用标准总线
    - Wishbone总线

#### □基本功能

- 数据存储、Seg7、SW、BTN和LED等接口
- □本实验用IP软核-U4
  - 核调用模块MIO\_BUS.ngc
  - 核接口信号模块(空文档): MIO\_BUS.v
  - 核模块符号文档: MIO\_BUS.sym

Peripheral\_in(31:0) GPIOe00000000 we BTN(3:0) GPIOf0000000\_we SW(15:0) mem\_w MIO BUS addr\_bus(31:0) —Cpu\_data4bus(31:0) led\_out(15:0)-— Cpu\_data2bus(31:0) ram\_data\_in(31:0) counter\_out(31:0) counter2\_out data\_ram\_we counter1\_out ram addr(9:0) counter0\_out ram\_data\_out(31:0) counter\_we

注: 如果增加RAM容量需修改地址译码

浙江大学 计算机学院 系统结构与系统软件实验室

# IO总线接口空模块-MIO\_BUS.v



```
module MIO_BUS( input wire clk, input wire rst,
                    input wire [3:0] BTN, input wire [15:0]SW,
                    input wire mem_w,
                    input wire [31:0] Cpu_data2bus,
                                                             //data from CPU
                    input wire [31:0] addr_bus,
                                                             //addr from CPU
                    input wire [31:0] ram_data_out,
                    input wire [15:0] led_out,
                    input wire [31:0] counter_out,
                    input wire counter0 out,
                    input wire counter1 out,
                    input wire counter2_out,
                     output wire [31:0] Cpu_data4bus,
                                                             //write to CPU
                     output wire [31:0] ram_data_in,
                                                             //from CPU write to Memory
                     output wire [9: 0] ram_addr,
                                                              //Memory Address signals
                     output wire data_ram_we,
                     output wire GPIOf0000000_w,
                                                             // GPIOffffff00 we
                     output wire GPIOe0000000 we,
                                                             // GPIOfffffe00 we
                                                             //记数器
                     output wire counter we,
                                                             //送外部设备总线
                     output wire [31:0] Peripheral in
                    );
```

# MIO\_BUS模块调用接口信号关系



#### **MIO\_BUS**

```
U4( clk_100HzM,
    rst],
    BTN [3:0],
    SW [15:0],
    mem_w,
    Cpu_data2bus [31:0],
    addr_bus [31:0],
    ram_data_out [31:0],
    led_out [7:0],
    counter_out [31:0],
    counter0_out,
    counter1_out,
    counter2_out,
```

```
Cpu_data4bus [31:0], ram_data_in [31:0], ram_addr [9: 0], data_ram_we, GPIOf0000000_w, GPIOe00000000_we, counter_we, Peripheral_in [31:0]
```

```
//主板时钟
//复位,按钮BTN3
//4位原始按钮输入
//8位原始开关输入
//存储器读写操作,来自CPU
//CPU输出数据总线
//地址总线,来自CPU
//来自RAM数据输出
//来自LED设备输出
//当前通道计数输出,来自计数器外设
//通道0计数结束输出,来自计数器外设
//通道1计数结束输出,来自计数器外设
//通道2计数结束输出,来自计数器外设
```

```
//CPU写入数据总线,连接到 CPU
//RAM 写入数据总线,连接到RAM
//RAM访问地址,连接到RAM
//RAM读写控制,连接到RAM
//设备一LED写信号
//设备二7段写信号,连接到U5
//记数器写信号,连接到U10
//外部设备写数据总线,连接所有写设备
```





# GPIO设备与接口模块

一非常简单与Exp03相同 -除Seg7设备外,接口与设备合二为一

浙江大学 计算机学院 系统结构与系统软件实验室

# U7-外部设备模块: GPIO接口及设备一 SPIO



#### □ GPIO输出设备一

- 地址范围=f0000000 fffffff0 (ffffff00-ffffffff0)
- 读写控制信号: GPIOf000000\_we(GPIOfffff00\_we)
- {GPIOf0[21:0],LED,counter\_set}

## □基本功能

- LEDs设备和计数器控制器读写
- 可回读,检测状态
- ■逻辑实验LED模块改造

#### □ 本实验用IP 软核- U7

- 核调用模块SPIO.ngc
- 核接口信号模块(空文档): SPIO\_IO.v
- 核模块符号文档: SPIO.sym



注意: 左移2位后低16位



计算机学院 系统结构与系统软件实验室

# 通用接口与设备一IP核调用空模块



#### -PIO.v

```
//io clk,与CPU反向
    module
                   PIO(input clk,
                       input rst,
                       input EN,
                                              //来自U4
                                              //来自U4
                       input [31:0] P_Data,
                                              //串行输出启动
                       input Start
            counter_set(1:0)-output [1:0] counter_set,//来自U7,后继用
clk
            LED_out(15:0) output [15:0] led_out,
                                              //输出到LED,回读到U4
rst
             GPIOf0(13:0)
                       output [13:0] GPIOf0
                                              //备用
- EN
                                              //串行时钟
                 led_sout output led_clk,
Start
                                              //串行LEDE值
                       output led_sout,
                       output LED_PED,
                                              //LED使能
                                              //LED清零
                       output led_clrn
                       );
    endmodule
```

# U6-外部设备模块: GPIO设备二

# THE UNITED

# SSeg\_Dev

#### □7段码显示输出设备模块

- 需要通过接口模块Multi\_8CH32与CPU连接
- 地址范围=E0000000 EFFFFFFF (FFFFE00-FFFFFFFF)

## □ 基本功能(参考Exp02)

- 4位7段码显示设备
- 模拟文本,显示8位16进制数: SW[1:0]=x1
  - □ SW[1:0]=01,显示低4位;
  - □ SW[1:0]=11,显示高4位

#### Sword平台不需要

- 模拟图形显示,4位7段用于32个点阵显示,SW[1:0]=x0
- 逻辑实验7段显示模块改造

## □本实验用IP 软核或Exp02设计的模块- U5

- 核调用模块SSeg\_Dev.ngc
- 核接口信号模块(空文档): SSeg\_Deg.v
- 核模块符号文档: SSeg\_Dev.sym



# U6-外部设备模块: GPIO设备二

Seg7\_Dev

SP3兼容adurinc

- □七段码显示输出设备模块
  - 需要通过接口模块Multi\_8CH32与CPU连接
  - 地址范围=E0000000 EFFFFFFF (FFFFE00-FFFFFFFF)
- □基本功能(参考OExp02)
  - 4位7段码显示设备
  - 模拟文本,显示8位16进制数: SW[1:0]=x1
    - □ SW[1:0]=01,显示低4位;
    - □ SW[1:0]=11,显示高4位
  - 模拟图形显示,4位7段用于32个点阵显示,SW[1:0]=x0
  - 逻辑实验7段显示模块改造
- □本实验用IP软核或Exp02设计的模块-U5
  - 核调用模块SSeg7\_Dev.ngc
  - 核接口信号模块(空文档): SSeg7\_Dev.v
  - 核模块符号文档: SSeg7\_Dev.sym

浙江大学 计算机学院 系统结构与系统软件实验室

# 通用设备二IP核调用空模块



### SSeg7\_Dev.v

```
//来自U5
module
          SSeg7_Dev(input wire [31:0] disp_num,
                        input wire [1:0]SW,
                                                            //来自U9
                                                            //通用分频器
                        input wire flash_clk,
                        input wire [1:0] Scanning, //来自U8
                        input wire [3:0] pointing, //来自U5
                                                            //来自U5
                        input wire [3:0] blinking,
                        output wire [3:0] AN,
                        output reg [7:0] SEGMENT //来自U5
                         );
endmodule
                                         clk
                                         rst
                                                 U6
                                                           seg_clk
                                         Start
                                         SW<sub>0</sub>
                                                          seg sout
                                                         SEG PEN
                                         flash
                                         Hexs(31:0)
                                                          seg clrn
                                         point(7:0)
                                         LES(7:0)
                                                    SSeg7_Dev
```

# U5-通用设备二接口模块

#### Multi\_8CH32



#### □ GPIO输出设备二接口模块

- 地址范围=E0000000 EFFFFFFF (FFFFE00-FFFFFFFF)
- 读写控制信号: GPIOe0000000\_we(GPIOffffe00\_we)

## □ 基本功能(参考Exp01)

- 七段码输出设备接口模块
- ■逻辑实验显示通道模块改造
- 通道0作为显示设备接口
  - **□ GPIOe0000000\_we=1**
  - □ CLK上升沿
- 通道1-7作为调试测试信号显示



### □本实验用IP软核或EXp01设计的模块-U6

- 核调用模块Multi\_8CH32.ngc
- 核接口信号模块(空文档): Multi\_8CH32\_IO.v
- 核模块符号文档: Multi\_8CH32.sym

# 通用设备二接口调用空模块 -Multi\_8CH32\_IO.v



| module          | Multi_8          |
|-----------------|------------------|
| - clk           |                  |
| - rst           | Disp_num(31:0) - |
| EN              | point_out(7:0) - |
| - Test(2:0)     | LE_out(7:0) -    |
| - point_in(63:0 | 0)               |
| LES(63:0)       |                  |
| Data0(31:0)     | U5               |
| data1(31:0)     | 03               |
| data2(31:0)     | 32               |
| data3(31:0)     | 当                |
| data4(31:0)     | 80               |
| - data5(31:0)   |                  |
| - data6(31:0)   | iii iii          |
| - data7(31:0)   | Σ                |
|                 |                  |

```
Multi_8CH32 (input clk,
                                      //io_clk,同步CPU
              input rst,
                                      //=1, 通道0显示
              input EN,
              input[63:0]point_in, //针对8个显示通道各8个小数点
              input[63:0]LE_in, //针对8个通道各8位闪烁控制
                                      //通道选择SW[7:5]
              input [2:0] Test,
                                      //通道0
              input [31:0] Data0,
                                      //通道1
              input [31:0] data0,
                                      //通道2
              input [31:0] data1,
                                      //通道3
              input [31:0] data2,
                                      //通道4
              input [31:0] data3,
              input [31:0] data4,
                                      //通道5
                                      //通道6
              input [31:0] data5,
                                      //通道7
              input [31:0] data6,
                                      //小数点输出
              output reg[7:0] point_out,
                                      //闪烁控制输出
              output reg[7:0] LE_out,
                                      //接入7段显示器
              output [31:0] Disp_num
            );
```

# Multi\_8CH32调用信号关系



```
Multi_8CH32
             U5(.clk(clk_io), .rst(rst),
                                                  //来自U4
                   . EN(GPIOe0000000_we),
                                                  //外部输入
                   . point_in(point_in),
                                                  //外部输入
                    .blink_in(LE_in),
                                                  //来自U9
                   .Test(SW_OK[7:5]),
                                                  //来自U4
                   .Data0(Peripheral_in),
                                                  //来自U1
                   .data1({2'b00,PC out[31:2]}),
                                                  //来自U10
                   .data2(counter_out),
                                                  //Inst, 来自CPU
                   .data3(Inst),
                                                   //来自CPU
                   .data4(addr bus),
                                                  //来自CPU
                   .data5(Cpu_data2bus),
                                                  //来自CPU
                   .data6(Cpu_data4bus),
                                                  //来自CPU;
                   .data7(PC_out),
                                                  //输出到U6
                   .point_out(point_out),
                                                  //输出到U6
                   .LE_out(blink_out),
                                                  //输出到U6
                   .disp_num(disp_num)
                   );
```

# 外部设备模块: GPIO接口设备三、四



GPIO\_SW\_BTN

### □ 15位Switch和4位Button输入设备

- 地址范围= f0000000-ffffffff0, A[2]=0
- 这二个设备非常简单直接包含在U4, MIO\_BUS模块中
- 与CPU数据线关系(当addre\_bus=f0000000时)

Cpu\_data4bus = {counter0\_out, counter1\_out, counter2\_out, led\_out[9:0], BTN,SW};注意: 低16位







计算机学院 系统结构与系统较

# U10-外部设备五:通用计数器模块

Counter\_x.v



#### □通用计数器设备,双向

- 地址范围=F0000004 FFFFFFF4 (FFFFFF04-FFFFFFF4)
- 读写控制信号: counter\_we

#### □基本功能

- 三通道独立计数器,可用于程序定时。
- 输出用于计数通道设置或计数值初始化
  - □ counter\_set=00、01、10对应计数通道0、1、2
  - □ counter\_set=11对应计数通道工作设置
- 计数器部分兼容8253

#### □ 本实验用IP 软核- U10

- 核调用模块Counter\_x.ngc
- 核接口信号模块(空文档): Counter\_x.v
- 核模块符号文档: Counter\_x.sym



浙江大学一计算

计算机学院 系统结构与系统软件实验室

# 通用计数器IP核调用空模块

# X.V

## -Counter\_x.v

```
module Counter_x(input clk,
                                          //io clk
                input rst,
                                          //Div[7],来自U8
                input clk0,
                                          //D], 来自U8
                input clk1,
                                          //Div[10], 来自U8
                input clk2,
                                          //计数器写控制,来自U4
                input counter_we,
                                       //计数器输入数据,来自U4
                input [31:0] counter_val,
                                        //计数器通道控制,来自U7
                input [1:0] counter_ch,
                                          //输出到U4
                output counter OUT,
                                           //输出到U4
                output counter1_OUT,
                                          //输出到U4
                output counter2_OUT,
                                          //输出到U4
                output [31:0] counter out
               );
```

endmodule





# SOC系统实现辅助模块

U8: 通用分频模块

U9: 开关去抖动模块

#### U8-通用分频模块: clk\_div



#### □计数分频模块

- ■用于要求不高的各类计数和分频
  - □CPU、IO和存储器等
- ■对延时和驱动有要求的需要BUFG缓冲
- ■对于时序要求高的需要用DCM实现

#### □基本功能

- 32位计数分频输出: Div
- CPU时钟输出: Clk CPU
- ■逻辑实验通用计数模块改造

#### □本实验自己设计核(逻辑电路输出)- U8

- 核调用模块clk\_div.ngc
- 核接口信号模块(空文档): clk\_div.v
- 核模块符号文档: clk\_div.sym

浙江大学 计算机学院 系统结构与系统软件实验室

#### 通用分频IP核调用空模块





endmodule



### U9-开关去抖动模块: SAnti\_jitter



#### □开关机械抖动消除模块

- ■用于消除开关和按钮输入信号的机械抖动
  - □CPU、IO和存储器等

#### □基本功能

- 输入机械开关量
- 输出滤除机械抖动的逻辑值
  - □ 电平输出: button\_out、SW\_OK
  - □ 脉冲输出: button\_pluse
- ■逻辑实验模块

#### □ 本实验可自己设计或用IP 软核- U9

- 核调用模块SAnti\_jitter.ngc
- 核接口信号模块(空文档): SAnti\_jitter.v
- 核模块符号文档: SAnti\_jitter.sym



沙大学 计算机学院 系统结构与系统软件实验室

#### 开关去抖动IP核调用空模块

#### -Anti\_jitter.v



module

SAnti\_jitter(input clk,

//主板时钟

RSTN CR — Clk U9 Key\_out(4:0) — Key\_y(3:0) Key\_ready — pulse\_out(3:0) — BTN\_OK(3:0) — SW(15:0) SW\_OK(15:0) — readn SAnti\_jitter rst

input RSTN //阵列式键盘读 input readn **input** [3:0]Key\_y, //阵列式键盘列输入 output reg[4:0] Key\_x, //阵列式键盘行输出 output reg[4:0] Key\_out,//阵列式键盘扫描码 output reg Key\_ready, //阵列式键盘有效 **input** [15:0] SW, //开关输入 output reg [3:0]] BTN\_OK,//列按键输出 output reg [3:0] pulse, //列按键脉冲输出 **output reg** [15:0] SW\_OK, //开关输出 **output reg** CR, //RSTN短按输出 output reg rst //复位, RSTN长按输出 );

endmodule



#### **Course Outline**



实验目的与实验环境

实验任务

实验原理

实验操作与实现

浙江大学系统结构与系统软件实验室

### 设计工程: OExp09-IP2MCPU



- ◎建立CPU调试、测试和应用环境
  - € 顶层用HDL实现,调用IP核模块
    - ⊙ 模块名: Top\_OExp09\_IP2MCPU.sch
- ◎SOC集成技术实现测试系统构架
  - € 复用实验三的模块,除SCPU外
    - CPU (第三方IP核): U1
    - ⊙RAM (ISE构建IP核): U3
    - 总线(第三方IP核): U4
    - 八数据通路模块(实验一Multi\_8CH32): U5
    - ⊙七段显示模块(实验二SSeg7\_Dev IP): U6
    - ⊙LED显示模块(实验二SPIO模块): U7
    - ⊙通用分频模块(clk\_div): U8
    - ⊙开关去抖模块(IP核): U9
    - ⊙数据输入模块(IP核): M4(目前没有使用)
- ◎ 分解CPU为二个IP核
  - ₠ SOC调试通过后用二个IP核构建MCPU



## 设计要点



## 建立多周期SOC测试应用环境

--新建工程OExp09-IP2MCPU

## 建立多周期SOC测试应用工程



#### □用ISE新建SOC测试应用工程

- 双击桌面上"Xilinx ISE"图标,启动ISE软件(也可从开始菜单启动)
- 选择File New Project选项,在弹出的对话框中输入工程名称并指定工程路径。参考工程名: ORG-Exp09或OExp09\_IP2MCPU
- 点击Next按钮进入下一页,选择所使用的芯片及综合、仿真工具。
- 再点击Next按钮进入下一页,这里显示了新建工程的信息,确认无误后,点击Finish就可以建立一个完整的工程了

#### □ 多周期CPU设计共享此工程



**New Project Wizard** Project Settings Specify device and project properties. Select the device and design flow for the project Value Property Name **Evaluation Development Board** None Specified Product Category v Family Spartan6 XC6SLX16 Device v CSG324 Package Speed Schematic Top-Level Source Type XST (VHDL/Verilog) Synthesis Tool v Simulator ISim (VHDL/Verilog) Preferred Language Verilog Property Specification in Project File Store all values v Manual Compile Order VHDL-93 VHDL Source Analysis Standard v Enable Message Filtering More Info < Back Cancel

## SOC测试应用框架工程模板





浙江大学一计算



#### 本实验采用结构化HDL描述不需要拷贝模块符号图

# 拷贝U1、U3~U10、多周期控制器、数据通路软核 (.ngc)文档到当前工程目录

一已经自己设计完成的除外 一通用分频模块建议采用自己设计的

## 设计多周期CPU核调用模块



#### □ 用HDL描述IP核调用实现CPU

■ 建议模块名: Multi\_CPU.v或MCPU.v



## 核集成CPU描述结构参考



```
module Muliti_CPU(input clk,
                                  //muliti CPU
             input reset,
             input MIO_ready,
             output[31:0] PC_out,
                                  //TEST
              output[31:0] inst_out,
                                  //TEST
              output mem_w,
             output[31:0] Addr_out,
              output[31:0] Data_out,
             input [31:0] Data_in,
             output CPU MIO,
             input INT,
             output[4:0]state
                                  //Test
             );
   ...........模块调用: 2个模块调用
endmodule
```

## 重要信号及模块调用结构



```
assign mem_w=MemWrite&&(~MemRead);
assign PC out=PC Current;
ctrl x_ctrl(.clk(clk),
          .reset(reset),
          .Inst in(inst out),
          .PCWrite(PCWrite),
          .PCWriteCond(PCWriteCond),
          .Branch(Branch)
          );
M_datapath x_datapath(.clk(clk),
                         .reset(reset),
                         .MIO_ready(MIO_ready),
                         .M addr(Addr out),
                         .zero(zero),
                         .overflow(overflow)
```



## SoC测试构架顶层结构描述

--用HDL实现,调用IP核模块

#### 根据下列逻辑原理图设计结构描述代码

很简单:就是端口定义和二级模块调用



浙江大学 计算机学院 系统结构与系统软件实验室

## 建立结构描述输入模板(顶层模块)

THE UNITED STATES

- □建立顶层模块
  - 在Project弹出的菜单中选择New Source命令
  - 选择: Verilog Module
  - 缺省目录是工程目录Exp09\_IP2MCPU
  - 建议修改为Exp09\_IP2CPU \Code
- □ 注意: 为了方便管理,将所有代码存放在独立目录中!

➤ New Source Wizard

←Select Source Type

Select source type, file name and its location.

- 同时注意同名.sch与.v文件的冲突
- □根据原理图设计描述代码
  - 后面以SOCMF.v为例说明



洲方子大学 Zhediana University

## 硬件描述代码输入窗口与环境





浙沙大学 计算机学院 系统结构与系统软件实验室

## 完成输入后第二层模块层次关系





### 顶层语法检查



#### □模块信号连接检测

- 激活Design窗口
- 点中顶层模块: SOCMF.v
- 在进程运行窗口: 选择Synthesze→Check Syntax
- ■综合器检查代码语法
  - □不会检查电路逻辑功能
  - □仅检查代码语法
  - □特别注意总线连接
    - 错位
    - ■别名





## 模块调用与关联



#### □模块调用方法

模块名 调用编号(端口信号列表);

- ■端口信号对应:
  - □与模块内端口信号顺序一一对应
  - □用括号引用: .模块内信号(输入信号)

如: .clk(clk\_100mHz)

#### □顶层调用模块关联

- 顶层窗口放置模块Symbol后会直接调用对应模块
- 建立核端口模块与软核模块关联
  - □只有端口信号的模块,没有逻辑代码的空文档.v
    - 综合器会根据端口模块连接信号
- 点击Add Source 关联对应的空模块

## 顶层(SOC测试应用系统)参考描述



```
module
         IP2CPU(input clk 50mhz,
                input [3:0]BTN,
                input [7:0]SW,
                output[7:0]LED,
                output[7:0]SEGMENT,
                output[3:0]AN_SEL
                );
Wire Clk_CPU, rst, clk_m, mem_w, data_ram_we, GPIOf0000000_we, GPIOe0000000_we, counter_we;
wire counter OUT0, counter OUT1, counter OUT2;
wire [1:0]Counter set;
wire MIO ready;
wire CPU_MIO;
      ............初值、模块连接过渡信号,信号转换、总线转换等
      ...........模块调用: 9个模块调用
```

endmodule



#### 信号传输:



#### 初值、模块连接过渡信号,信号转换、总线转换等





```
//主板时钟
SAnti_jitter
           U9(clk,
              RSTN
              readn
                             //阵列式键盘列输入
              Key_y,
                             //阵列式键盘行输出
              Key_x,
                             //阵列式键盘扫描码
              Key_out,
                             //复位, RSTN长按输出
               rst
               );
                             // Clock divider-时钟分频器
clk_div
           U8(clk_50mhz,
              rst,
              SW2,
              clkdiv,
              Clk_CPU
              );
```



#### SSeg7\_Dev



#### Muliti\_CPU

```
U1(.clk(Clk_CPU),
.reset(rst),
.MIO_ready(MIO_ready),
.....
.state(state)
);
```



//MIO\_ready

//Test



```
RAM_B
                               U3(.clka(clk_m),
                                     .wea(data_ram_we),
                                     .addra(ram_addr),
                                                                           // Addre_Bus [9 : 0]
addra(9:0)
             U3
                                     .dina(ram_data_in),
                      douta(31:0)
wea(0:0)
                                     .douta(ram_data_out)
                                                                           //Data_Bus [31 : 0]
-d in a(31:0)
           RAM B
                                     );
 clka
                                                                                                   Peripheral in(31:0)
                                                                                                  GPIOe0000000 we
                                                                                    BTN(3:0)
                                                                                                   GPIOf0000000_we
            MIO_BUS
                               U4(.clk(clk_50mhz),
                                                                                    SW(15:0)
                                                                                             MIO BUS
                                                                                    - addr_bus(31:0)
                                                                                    —Cpu_data4bus(31:0)
                                                                                                       led_out(15:0)-
                                                                                    Cpu_data2bus(31:0)
                                                                                      ram_data_in(31:0)
                                     .Peripheral_in(Peripheral_in)
                                                                                                    counter_out(31:0)
                                                                                    data_ram_we
                                                                                                       counter2 out
                                     );
                                                                                                       counter1_out
                                                                                    ram addr(9:0)
                                                                                                       counter0 out
                                                                                    ram_data_out(31:0)
                                                                                                        counter we
```





```
SPIO_IO
                               U7( clk,
                                                                clk
                                                                                   counter_set(1:0)
                                                                                     LED_out(15:0)
                                    GPIO<sub>f</sub>0
                                                                rst
                                                                                      GPIOf0(13:0)
                                                                 EN
                                                                                             led clk
                                    );
                                                                                            led sout
                                                                Start
                                                                                            LED_PEN
                                                                P_Data(31:0)
      Multi_8CH32
                             U5(.clk(??),
                                   .rst(rst),
clk
                                   .EN(??),
        Disp_num(31:0) -
rst
                                   .\text{Test}(SW\_OK[7:5]),
EN
          point_out(7:0)
Test(2:0)
           LE out(7:0)
                                   .Data0(??),
                                                                    //CPU data output
                                   .data1(??),
                                                                   //pc[31:2]
point_in(63:0)
LES(63:0)
                                   .data2(counter_out),
                                                                   //counter
Data0(31:0)
             U<sub>5</sub>
                                   .data3(??),
                                                                   //Inst
data1(31:0)
                                   .data4(??),
                                                                   //addr bus
data2(31:0)
                                                                   //Cpu_data2bus;
                                   .data5(??),
data3(31:0)
                                                                   //Cpu_data4bus;
                                   .data6(??),
data4(31:0)
                                   .data6(??),
                                                                   //pc;
data5(31:0)
data6(31:0)
data7(31:0)
```

浙江大学



#### Counter\_x

#### **U10**(.clk(clk\_io),



```
.clk1(clkdiv[10]),
```

.clk0(clkdiv[9]),

.clk2(clkdiv[10]),

.counter\_we(counter\_we),

.counter\_val(Peripheral\_in),

.counter\_ch(Counter\_set),

.counter0\_OUT(counter\_OUT0),

.counter1\_OUT(counter\_OUT1),

.counter2\_OUT(counter\_OUT2),

.counter\_out(counter\_out)

**)**;

endmodule



## 完成后的层次关联







# SoC物理调试验证

-本实验不需要测试,仅用DEMO程序验证功能 -过程与实验三、四相同

## 物理验证-DEMO接口功能



没有使用



SW[7:5]=显示通道选择

SW[7:5]=000: CPU程序运行输出

ISW[7:5]=001:测试PC字地址

SW[7:5]=010:测试指令字

SW[7:5]=011: 测试计数器

SW[7:5]=100: 测试RAM地址

SW[7:5]=101:测试CPU数据输出

【SW[7:5]=110:测试CPU数据输入

SW[0]=文本图形选择

SW[1]=高低16位选择

SW[2]=CPU单步时钟选择

SW[4:3]=00,点阵显示程序:跑马灯

SW[4:3]=00,点阵显示程序:矩形变幻

【SW[4:3]=01,内存数据显示程序: 0~F

SW[4:3]=10,当前寄存器+1显示



### 下载验证SoC



- □非IP核仿真
  - 对自己设计的模块做时序仿真(单周期时仿真过的略)
  - 第三方IP核不做仿真(固核无法做仿真)
- □SOC物理验证
  - 下载流文件.bit
  - 验证调试SOC功能
    - □功能不正确时排查错误
  - 定性观测SOC关键信号
    - □本实验只要求定性观测
    - □测试代码和数据用mem.coe数据\*

## 测试开关设置



#### □图形功能测试

| 开关      | 位置  | 功能                      |
|---------|-----|-------------------------|
| SW[1:0] | X0  | 七段码图形显示                 |
| SW[2]   | 0   | CPU全速时钟                 |
| SW[4:3] | 00  | 7段码从上至下亮点循环右移           |
| SW[4:3] | 11  | 7段码矩形从下到大循环显示           |
| SW[7:5] | 000 | 作为外设使用(E000000/FFFFE00) |

#### □文本功能测试

| 开关      | 位置  | 功能                      |
|---------|-----|-------------------------|
| SW[1:0] | 01  | 七段码文本显示(低16位)           |
|         | 11  | 七段码文本显示(高16位)           |
| SW[2]   | 0   | CPU全速时钟                 |
| SW[4:3] | 01  | 7段码显示RAM数字              |
| SW[4:3] | 10  | 7段码显示累加                 |
| SW[7:5] | 000 | 作为外设使用(E000000/FFFFE00) |

浙沙大学 计算机学院 系统结构与系统软件实验室

## 仅定性观测



#### □SOC信号测试

- CPU全速运行
- ■测试开关设置

| 开关      | 位置  | 功能                         |
|---------|-----|----------------------------|
| SW[1:0] | 01  | 七段码文本显示(低16位)              |
| SW[1:0] | 11  | 七段码文本显示(高16位)              |
| SW[2]   | 0   | CPU全速时钟                    |
| SW[7:5] | 010 | Counter值输出                 |
| SW[7:5] | 100 | CPU数据存储地址addr_bus(ALU)     |
| SW[7:5] | 101 | CPU数据输出Cpu_data2bus (寄存器B) |
| SW[7:5] | 110 | CPU数据输入Cpu_data4bus(RAM输出) |

## 仅定性观测



#### □SOC信号测试

- CPU单步运行
- ■测试开关设置
- 设计测试程序替换DEMO程序\*

| 开关      | 位置  | 功能                         |
|---------|-----|----------------------------|
| SW[1:0] | 01  | 七段码文本显示(低16位)              |
| SW[1:0] | 11  | 七段码文本显示(高16位)              |
| SW[2]   | 1   | CPU单步时钟                    |
| SW[7:5] | 001 | CPU指令字地址PC_out[31:2]       |
| SW[7:5] | 011 | ROM指令输出Inst_in             |
| SW[7:5] | 100 | CPU数据存储地址addr_bus(ALU输出)   |
| SW[7:5] | 101 | CPU数据输出Cpu_data2bus(寄存器B)  |
| SW[7:5] | 110 | CPU数据输入Cpu_data4bus(RAM输出) |
| SW[7:5] | 111 | CPU指令字节地址PC_out            |



# OEND)